home *** CD-ROM | disk | FTP | other *** search
/ Amiga Tools 5 / Amiga Tools 5.iso / tools / developer-tools / andere sprachen / oberonv4 / oberon-src / nonfpu / texts.mod (.txt) < prev   
Encoding:
Oberon Text  |  1995-11-22  |  32.4 KB  |  874 lines

  1. Syntax10.Scn.Fnt
  2. ParcElems
  3. Alloc
  4. Syntax24b.Scn.Fnt
  5. StampElems
  6. Alloc
  7. 22 Nov 95
  8. Syntax10b.Scn.Fnt
  9. (* AMIGA NonFPU*)
  10. MODULE Texts;    (** CAS/HM 23.9.93 -- interface based on Texts by JG/NW 6.12.91**)
  11.     (* << RC, MB, JT, SHML, CN 
  12.     IMPORT
  13.         Files, Modules, Fonts, Display, Reals(* <<, X11*), AmigaMath, AmigaMathL;
  14.     (*--- insert field e: Elem into Texts.Scanner and change Texts.Scan to set it in case of class=6 *)
  15.     CONST
  16.         ElemChar* = 1CX;
  17.         TAB = 9X; CR = 0DX; maxD = 9;
  18.         (**FileMsg.id**)
  19.             load* = 0; store* = 1;
  20.         (**Notifier op**)
  21.             replace* = 0; insert* = 1; delete* = 2;
  22.         (**Scanner.class**)
  23.             Inval* = 0; Name* = 1; String* = 2; Int* = 3; Real* = 4; LongReal* = 5; Char* = 6;
  24.         textTag = 0F0X; version = 01X;
  25.     TYPE
  26.         Run = POINTER TO RunDesc;
  27.         RunDesc = RECORD
  28.             prev, next: Run;
  29.             len: LONGINT;
  30.             fnt: Fonts.Font;
  31.             col, voff: SHORTINT;
  32.             ascii: BOOLEAN    (* << *)
  33.         END;
  34.         Piece = POINTER TO PieceDesc;
  35.         PieceDesc = RECORD (RunDesc)
  36.             file: Files.File;
  37.             org: LONGINT
  38.         END;
  39.         Elem* = POINTER TO ElemDesc;
  40.         Buffer* = POINTER TO BufDesc;
  41.         Text* = POINTER TO TextDesc;
  42.         ElemMsg* = RECORD END;
  43.         Handler* = PROCEDURE (e: Elem; VAR msg: ElemMsg);
  44.         ElemDesc* = RECORD (RunDesc)
  45.             W*, H*: LONGINT;
  46.             handle*: Handler;
  47.             base: Text
  48.         END;
  49.         FileMsg* = RECORD (ElemMsg)
  50.             id*: INTEGER;
  51.             pos*: LONGINT;
  52.             r*: Files.Rider
  53.         END;
  54.         CopyMsg* = RECORD (ElemMsg)
  55.             e*: Elem
  56.         END;
  57.         IdentifyMsg* = RECORD (ElemMsg)
  58.             mod*, proc*: ARRAY 32 OF CHAR
  59.         END;
  60.         BufDesc* = RECORD
  61.             len*: LONGINT;
  62.             head: Run
  63.         END;
  64.         Notifier* = PROCEDURE (T: Text; op: INTEGER; beg, end: LONGINT);
  65.         TextDesc* = RECORD
  66.             len*: LONGINT;
  67.             notify*: Notifier;
  68.             head, cache: Run;
  69.             corg: LONGINT
  70.         END;
  71.         Reader* = RECORD
  72.             eot*: BOOLEAN;
  73.             fnt*: Fonts.Font;
  74.             col*, voff*: SHORTINT;
  75.             elem*: Elem;
  76.             rider: Files.Rider;
  77.             run: Run;
  78.             org, off: LONGINT
  79.         END;
  80.         Scanner* = RECORD (Reader)
  81.             nextCh*: CHAR;
  82.             line*, class*: INTEGER;
  83.             i*: LONGINT;
  84.             x*: REAL;
  85.             y*: LONGREAL;
  86.             c*: CHAR;
  87.             len*: SHORTINT;
  88.             s*: ARRAY 64 OF CHAR    (* << *)
  89.         END;
  90.         Writer* = RECORD
  91.             buf*: Buffer;
  92.             fnt*: Fonts.Font;
  93.             col*, voff*: SHORTINT;
  94.             rider: Files.Rider;
  95.             file: Files.File
  96.         END;
  97.         Alien = POINTER TO RECORD (ElemDesc)
  98.             file: Files.File;
  99.             org, span: LONGINT;
  100.             mod, proc: ARRAY 32 OF CHAR
  101.         END;
  102.         new*: Elem;
  103.         del: Buffer;
  104.     (* run primitives *)
  105. (* for FPU-Code use Ten and TenL from Reals *)
  106. PROCEDURE Ten(e: INTEGER; VAR m: REAL);
  107.     VAR r, power: LONGREAL;
  108. BEGIN
  109.     r := 1;
  110.     power := 10;
  111.     WHILE e > 0 DO
  112.         IF ODD(e) THEN AmigaMathL.Mul(r, power, r) END;
  113.         AmigaMathL.Mul(power, power, power); e := e DIV 2
  114.     END ;
  115.     AmigaMathL.Short(r, m);
  116. END Ten;
  117. PROCEDURE TenL(e: INTEGER; VAR r: LONGREAL);
  118.  VAR power: LONGREAL;
  119. BEGIN r := 1;
  120.  power := 10;
  121.  LOOP
  122.   IF ODD(e) THEN AmigaMathL.Mul(r, power, r)END ;
  123.   e := e DIV 2;
  124.   IF e <= 0 THEN RETURN END ;
  125.   AmigaMathL.Mul(power, power, power)
  126. END TenL;
  127.     PROCEDURE Find (T: Text; VAR pos: LONGINT; VAR u: Run; VAR org, off: LONGINT);
  128.         VAR v: Run; m: LONGINT;
  129.     BEGIN
  130.         IF pos >= T.len THEN pos := T.len; u := T.head; org := T.len; off := 0; T.cache := T.head; T.corg := 0
  131.         ELSE v := T.cache.next; m := pos - T.corg;
  132.             IF pos >= T.corg THEN
  133.                 WHILE m >= v.len DO DEC(m, v.len); v := v.next END
  134.             ELSE
  135.                 WHILE m < 0 DO v := v.prev; INC(m, v.len) END
  136.             END;
  137.             u := v; org := pos - m; off := m; T.cache := v.prev; T.corg := org
  138.         END
  139.     END Find;
  140.     PROCEDURE Split (off: LONGINT; VAR u, un: Run);
  141.         VAR p, U: Piece;
  142.     BEGIN
  143.         IF off = 0 THEN un := u; u := un.prev
  144.         ELSIF off >= u.len THEN un := u.next
  145.         ELSE NEW(p); un := p; U := u(Piece);
  146.             p^ := U^; INC(p.org, off); DEC(p.len, off); DEC(U.len, p.len);
  147.             p.ascii := u.ascii; p.prev := U; p.next := U.next; p.next.prev := p; U.next := p    (* << *)
  148.         END
  149.     END Split;
  150.     PROCEDURE Merge (T: Text; u: Run; VAR v: Run);
  151.         VAR p, q: Piece;
  152.     BEGIN
  153.         IF (u IS Piece) & (v IS Piece) & (u.fnt.name = v.fnt.name) & (u.col = v.col) & (u.voff = v.voff)
  154.         & (u(Piece).ascii = v(Piece).ascii) THEN    (* << *)
  155.             p := u(Piece); q := v(Piece);
  156.             IF (p.file = q.file) & (p.org + p.len = q.org) THEN
  157.                 IF T.cache = u THEN INC(T.corg, q.len)
  158.                 ELSIF T.cache = v THEN T.cache := T.head; T.corg := 0
  159.                 END;
  160.                 INC(p.len, q.len); v := v.next
  161.             END
  162.         END
  163.     END Merge;
  164.     PROCEDURE Splice (un, v, w: Run; base: Text);    (* (u, un) -> (u, v, w, un) *)
  165.         VAR u: Run;
  166.     BEGIN
  167.         IF v # w.next THEN u := un.prev;
  168.             u.next := v; v.prev := u; un.prev := w; w.next := un;
  169.             REPEAT
  170.                 IF v IS Elem THEN v(Elem).base := base END;
  171.                 v := v.next
  172.             UNTIL v = un
  173.         END
  174.     END Splice;
  175.     PROCEDURE ClonePiece (p: Piece): Piece;
  176.         VAR q: Piece;
  177.     BEGIN NEW(q); q^ := p^; RETURN q
  178.     END ClonePiece;
  179.     PROCEDURE CloneElem (e: Elem): Elem;
  180.         VAR msg: CopyMsg;
  181.     BEGIN msg.e := NIL; e.handle(e, msg); RETURN msg.e
  182.     END CloneElem;
  183.     (** Elements **)
  184.     PROCEDURE CopyElem* (SE, DE: Elem);
  185.     BEGIN DE.len := SE.len; DE.fnt := SE.fnt; DE.col := SE.col; DE.voff := SE.voff;
  186.         DE.W := SE.W; DE.H := SE.H; DE.handle := SE.handle
  187.     END CopyElem;
  188.     PROCEDURE ElemBase* (E: Elem): Text;
  189.     BEGIN RETURN E.base
  190.     END ElemBase;
  191.     PROCEDURE ElemPos* (E: Elem): LONGINT;
  192.         VAR u: Run; pos: LONGINT;
  193.     BEGIN u := E.base.head.next; pos := 0;
  194.         WHILE u # E DO pos := pos + u.len; u := u.next END;
  195.         RETURN pos
  196.     END ElemPos;
  197.     PROCEDURE HandleAlien (E: Elem; VAR msg: ElemMsg);
  198.         VAR e: Alien; r: Files.Rider; i: LONGINT; ch: CHAR;
  199.     BEGIN
  200.         WITH E: Alien DO
  201.             IF msg IS CopyMsg THEN
  202.                 WITH msg: CopyMsg DO NEW(e); CopyElem(E, e);
  203.                     e.file := E.file; e.org := E.org; e.span := E.span; e.mod := E.mod; e.proc := E.proc;
  204.                     msg.e := e
  205.                 END
  206.             ELSIF msg IS IdentifyMsg THEN
  207.                 WITH msg: IdentifyMsg DO
  208.                     COPY(E.mod, msg.mod); COPY(E.proc, msg.proc); msg.mod[31] := 1X (*alien*)
  209.                 END
  210.             ELSIF msg IS FileMsg THEN
  211.                 WITH msg: FileMsg DO
  212.                     IF msg.id = store THEN Files.Set(r, E.file, E.org); i := E.span;
  213.                         WHILE i > 0 DO Files.Read(r, ch); Files.Write(msg.r, ch); DEC(i) END
  214.                     END
  215.                 END
  216.             END
  217.         END
  218.     END HandleAlien;
  219.     (** Buffers **)
  220.     PROCEDURE OpenBuf* (B: Buffer);
  221.         VAR u: Run;
  222.     BEGIN NEW(u);  u.next := u;  u.prev := u; B.head := u; B.len := 0
  223.     END OpenBuf;
  224.     PROCEDURE Copy* (SB, DB: Buffer);
  225.         VAR u, v, vn: Run;
  226.     BEGIN u := SB.head.next; v := DB.head.prev;
  227.         WHILE u # SB.head DO
  228.             IF u IS Piece THEN vn := ClonePiece(u(Piece)) ELSE vn := CloneElem(u(Elem)) END;
  229.             v.next := vn; vn.prev := v; v := vn; u := u.next
  230.         END;
  231.         v.next := DB.head; DB.head.prev := v;
  232.         INC(DB.len, SB.len)
  233.     END Copy;
  234.     PROCEDURE Recall* (VAR B: Buffer);
  235.     BEGIN B := del; del := NIL
  236.     END Recall;
  237.     (** Texts **)
  238.     PROCEDURE Save* (T: Text; beg, end: LONGINT; B: Buffer);
  239.         VAR u, v, w, wn: Run; uo, ud, vo, vd: LONGINT;
  240.     BEGIN Find(T, beg, u, uo, ud); Find(T, end, v, vo, vd);
  241.         w := B.head.prev;
  242.         WHILE u # v DO
  243.             IF u IS Piece THEN wn := ClonePiece(u(Piece)); DEC(wn.len, ud); INC(wn(Piece).org, ud)
  244.             ELSE wn := CloneElem(u(Elem))
  245.             END;
  246.             w.next := wn; wn.prev := w; w := wn; u := u.next; ud := 0
  247.         END;
  248.         IF vd > 0 THEN (*v IS Piece*) wn := ClonePiece(v(Piece)); wn.len := vd - ud; INC(wn(Piece).org, ud);
  249.             w.next := wn; wn.prev := w; w := wn
  250.         END;
  251.         w.next := B.head; B.head.prev := w;
  252.         INC(B.len, end - beg)
  253.     END Save;
  254.     PROCEDURE Insert* (T: Text; pos: LONGINT; B: Buffer);
  255.         VAR u, un, v: Run; p, q: Piece; uo, ud, len: LONGINT;
  256.     BEGIN Find(T, pos, u, uo, ud); Split(ud, u, un);
  257.         len := B.len; v := B.head.next;
  258.         Merge(T, u, v); Splice(un, v, B.head.prev, T);
  259.         INC(T.len, len); B.head.next := B.head; B.head.prev := B.head; B.len := 0;
  260.         T.notify(T, insert, pos, pos + len);
  261.         (*X11.DoFlush     << *)
  262.     END Insert;
  263.     PROCEDURE Append* (T: Text; B: Buffer);
  264.         VAR v: Run; pos, len: LONGINT;
  265.     BEGIN pos := T.len; len := B.len; v := B.head.next;
  266.         Merge(T, T.head.prev, v); Splice(T.head, v, B.head.prev, T);
  267.         INC(T.len, len); B.head.next := B.head; B.head.prev := B.head; B.len := 0;
  268.         T.notify(T, insert, pos, pos + len);
  269.         (*X11.DoFlush     << *)
  270.     END Append;
  271.     PROCEDURE Delete* (T: Text; beg, end: LONGINT);
  272.         VAR c, u, un, v, vn: Run; co, uo, ud, vo, vd: LONGINT;
  273.     BEGIN
  274.         Find(T, beg, u, uo, ud); Split(ud, u, un); c := T.cache; co := T.corg;
  275.         Find(T, end, v, vo, vd); Split(vd, v, vn); T.cache := c; T.corg := co;
  276.         NEW(del); OpenBuf(del); del.len := end - beg;
  277.         Splice(del.head, un, v, NIL);
  278.         Merge(T, u, vn); u.next := vn; vn.prev := u;
  279.         DEC(T.len, end - beg);
  280.         T.notify(T, delete, beg, end);
  281.     END Delete;
  282.     PROCEDURE ChangeLooks* (T: Text; beg, end: LONGINT; sel: SET; fnt: Fonts.Font; col, voff: SHORTINT);
  283.         VAR c, u, un, v, vn: Run; co, uo, ud, vo, vd: LONGINT;
  284.     BEGIN Find(T, beg, u, uo, ud); Split(ud, u, un); c := T.cache; co := T.corg;
  285.         Find(T, end, v, vo, vd); Split(vd, v, vn); T.cache := c; T.corg := co;
  286.         WHILE un # vn DO
  287.             IF (0 IN sel) & (fnt # NIL) THEN un.fnt := fnt END;
  288.             IF 1 IN sel THEN un.col := col END;
  289.             IF 2 IN sel THEN un.voff := voff END;
  290.             Merge(T, u, un);
  291.             IF u.next = un THEN u := un; un := un.next ELSE u.next := un; un.prev := u END
  292.         END;
  293.         Merge(T, u, un); u.next := un; un.prev := u;
  294.         T.notify(T, replace, beg, end);
  295.     END ChangeLooks;
  296.     (** Readers **)
  297.     PROCEDURE OpenReader* (VAR R: Reader; T: Text; pos: LONGINT);
  298.         VAR u: Run;
  299.     BEGIN
  300.         IF pos >= T.len THEN pos := T.len END;
  301.         Find(T, pos, u, R.org, R.off); R.run := u; R.eot := FALSE;
  302.         IF u IS Piece THEN
  303.             Files.Set(R.rider, u(Piece).file, u(Piece).org + R.off)
  304.         END
  305.     END OpenReader;
  306.     PROCEDURE Read* (VAR R: Reader; VAR ch: CHAR);
  307.         VAR u: Run;
  308.     BEGIN u := R.run; R.fnt := u.fnt; R.col := u.col; R.voff := u.voff; INC(R.off);
  309.         IF u IS Piece THEN Files.Read(R.rider, ch); R.elem := NIL;
  310.             IF (ch = 0AX) & u(Piece).ascii THEN ch := CR END    (* << LF to CR *)
  311.         ELSIF u IS Elem THEN ch := ElemChar; R.elem := u(Elem)
  312.         ELSE ch := 0X; R.elem := NIL; R.eot := TRUE
  313.         END;
  314.         IF R.off = u.len THEN INC(R.org, u.len); u := u.next;
  315.             IF u IS Piece THEN
  316.                 WITH u: Piece DO Files.Set(R.rider, u.file, u.org) END
  317.             END;
  318.             R.run := u; R.off := 0
  319.         END
  320.     END Read;
  321.     PROCEDURE ReadElem* (VAR R: Reader);
  322.         VAR u, un: Run;
  323.     BEGIN u := R.run;
  324.         WHILE u IS Piece DO INC(R.org, u.len); u := u.next END;
  325.         IF u IS Elem THEN un := u.next; R.run := un; INC(R.org); R.off := 0;
  326.             R.fnt := u.fnt; R.col := u.col; R.voff := u.voff; R.elem := u(Elem);
  327.             IF un IS Piece THEN
  328.                 WITH un: Piece DO Files.Set(R.rider, un.file, un.org) END
  329.             END
  330.         ELSE R.eot := TRUE; R.elem := NIL
  331.         END
  332.     END ReadElem;
  333.     PROCEDURE ReadPrevElem* (VAR R: Reader);
  334.         VAR u: Run;
  335.     BEGIN u := R.run.prev;
  336.         WHILE u IS Piece DO DEC(R.org, u.len); u := u.prev END;
  337.         IF u IS Elem THEN R.run := u; DEC(R.org); R.off := 0;
  338.             R.fnt := u.fnt; R.col := u.col; R.voff := u.voff; R.elem := u(Elem)
  339.         ELSE R.eot := TRUE; R.elem := NIL
  340.         END
  341.     END ReadPrevElem;
  342.     PROCEDURE Pos* (VAR R: Reader): LONGINT;
  343.     BEGIN RETURN R.org + R.off
  344.     END Pos;
  345.     (** Scanners --------------- NW --------------- **)
  346.     PROCEDURE OpenScanner* (VAR S: Scanner; T: Text; pos: LONGINT);
  347.     BEGIN OpenReader(S, T, pos); S.line := 0; S.nextCh := " "
  348.     END OpenScanner;
  349.     (*IEEE floating point formats:
  350.         x = 2^(e-127) * 1.m    bit 0: sign, bits 1- 8: e, bits  9-31: m
  351.         x = 2^(e-1023) * 1.m   bit 0: sign, bits 1-11: e, bits 12-63: m *)
  352.     PROCEDURE Scan* (VAR S: Scanner);
  353.         CONST maxD = 32;
  354.         VAR ch, term: CHAR;
  355.             neg, negE, hex: BOOLEAN;
  356.             i, j, h: SHORTINT;
  357.             e: INTEGER; k: LONGINT;
  358.             x, f: REAL; y, g: LONGREAL; Dum: REAL; DumL: LONGREAL;
  359.             d: ARRAY maxD OF CHAR;
  360.         PROCEDURE ReadScaleFactor;
  361.         BEGIN Read(S, ch);
  362.             IF ch = "-" THEN negE := TRUE; Read(S, ch)
  363.             ELSE negE := FALSE;
  364.                 IF ch = "+" THEN Read(S, ch) END
  365.             END;
  366.             WHILE ("0" <= ch) & (ch <= "9") DO
  367.                 e := e*10 + ORD(ch) - 30H; Read(S, ch)
  368.             END
  369.         END ReadScaleFactor;
  370.     BEGIN ch := S.nextCh; i := 0;
  371.         LOOP
  372.             IF ch = CR THEN INC(S.line)
  373.             ELSIF (ch # " ") & (ch # TAB) THEN EXIT
  374.             END ;
  375.             Read(S, ch)
  376.         END;
  377.         IF ("A" <= CAP(ch)) & (CAP(ch) <= "Z") OR (ch = "/") OR (ch = ":") OR (ch = ".") THEN (*name*)    (* << *)
  378.             REPEAT S.s[i] := ch; INC(i); Read(S, ch)
  379.             UNTIL ("Z" < CAP(ch)) & (ch # "_")    (* << *)
  380.                 OR ("9" < ch) & (CAP(ch) < "A") & (ch # ":")    (* << *)
  381.                 OR (ch < "0") & (ch # ".") & (ch # "/")    (* << *)
  382.                 OR (i = 63);    (* << *)
  383.             S.s[i] := 0X; S.len := i; S.class := 1
  384.         ELSIF ch = 22X THEN (*literal string*)
  385.             Read(S, ch);
  386.             WHILE (ch # 22X) & (ch >= " ") & (i # 63) DO    (* << *)
  387.                 S.s[i] := ch; INC(i); Read(S, ch)
  388.             END;
  389.             S.s[i] := 0X; S.len := i; Read(S, ch); S.class := 2
  390.         ELSE
  391.             IF ch = "-" THEN neg := TRUE; Read(S, ch) ELSE neg := FALSE END ;
  392.             IF ("0" <= ch) & (ch <= "9") THEN (*number*)
  393.                 hex := FALSE; j := 0;
  394.                 LOOP d[i] := ch; INC(i); Read(S, ch);
  395.                     IF ch < "0" THEN EXIT END;
  396.                     IF "9" < ch THEN
  397.                         IF ("A" <= ch) & (ch <= "F") THEN hex := TRUE; ch := CHR(ORD(ch)-7)
  398.                         ELSIF ("a" <= ch) & (ch <= "f") THEN hex := TRUE; ch := CHR(ORD(ch)-27H)
  399.                         ELSE EXIT
  400.                         END
  401.                     END
  402.                 END;
  403.                 IF ch = "H" THEN (*hex number*)
  404.                     Read(S, ch); S.class := 3;
  405.                     IF i-j > 8 THEN j := i-8 END ;
  406.                     k := ORD(d[j]) - 30H; INC(j);
  407.                     IF (i-j = 7) & (k >= 8) THEN DEC(k, 16) END ;
  408.                     WHILE j < i DO k := k*10H + (ORD(d[j]) - 30H); INC(j) END ;
  409.                     IF neg THEN S.i := -k ELSE S.i := k END
  410.                 ELSIF ch = "." THEN (*read real*)
  411.                     Read(S, ch); h := i;
  412.                     WHILE ("0" <= ch) & (ch <= "9") DO d[i] := ch; INC(i); Read(S, ch) END ;
  413.                     IF ch = "D" THEN
  414.                         e := 0; y := 0; g := 1;
  415.                         REPEAT
  416.                             AmigaMathL.IntToReal(ORD(d[j]) - 30H, DumL);
  417.                             AmigaMathL.Mul(y, 10, y);
  418.                             AmigaMathL.Add(y, DumL, y);
  419.                             (* y := y*10 + (ORD(d[j]) - 30H);*)
  420.                             INC(j)
  421.                         UNTIL j = h;
  422.                         WHILE j < i DO
  423.                             AmigaMathL.Div(g, 10, g);
  424.                             AmigaMathL.IntToReal(ORD(d[j]) - 30H, DumL);
  425.                             AmigaMathL.Mul(DumL, g, DumL);
  426.                             AmigaMathL.Add(DumL, y, y);
  427.                             (*g := g/10; y := (ORD(d[j]) - 30H)*g + y;*)
  428.                             INC(j)
  429.                         END ;
  430.                         ReadScaleFactor;
  431.                         IF negE THEN
  432.                             IF e <= 308 THEN
  433.                                 TenL(e, DumL);
  434.                                 AmigaMathL.Div(y, DumL, y);
  435.                                 (* y := y / Reals.TenL(e)*)
  436.                             ELSE y := 0 END
  437.                         ELSIF e > 0 THEN
  438.                             IF e <= 308 THEN
  439.                                 TenL(e, DumL);
  440.                                 AmigaMathL.Mul(y, DumL, y);
  441.                                 (* y := Reals.TenL(e) * y;*)
  442.                             ELSE HALT(40) END
  443.                         END ;
  444.                         IF neg THEN
  445.                             AmigaMathL.Neg(y, y);
  446.                             (* y := -y *)
  447.                         END ;
  448.                         S.class := 5; S.y := y
  449.                     ELSE e := 0; x := 0; f := 1;
  450.                         REPEAT
  451.                             AmigaMath.IntToReal(ORD(d[j]) - 30H, Dum);
  452.                             AmigaMath.Mul(x, 10, x);
  453.                             AmigaMath.Add(x, Dum, x);
  454.                             (* x := x*10 + (ORD(d[j]) - 30H);*)
  455.                             INC(j)
  456.                         UNTIL j = h;
  457.                         WHILE j < i DO
  458.                             AmigaMath.Div(f, 10, f);
  459.                             AmigaMath.IntToReal(ORD(d[j])-30H, Dum);
  460.                             AmigaMath.Mul(Dum, f, Dum);
  461.                             AmigaMath.Add(Dum, x, x);
  462.                             (*f := f/10; x := (ORD(d[j])-30H)*f + x;*)
  463.                             INC(j)
  464.                         END;
  465.                         IF ch = "E" THEN ReadScaleFactor END ;
  466.                         IF negE THEN
  467.                             IF e <= 38 THEN
  468.                                 Ten(e, Dum);
  469.                                 AmigaMath.Div(x, Dum, x);
  470.                                 (*x := x / Reals.Ten(e)*)
  471.                             ELSE x := 0 END
  472.                         ELSIF e > 0 THEN
  473.                             IF e <= 38 THEN
  474.                                 Ten(e, Dum);
  475.                                 AmigaMath.Mul(Dum, x, x);
  476.                                 (*x := Reals.Ten(e) * x*)
  477.                             ELSE HALT(40) END
  478.                         END ;
  479.                         IF neg THEN
  480.                             AmigaMath.Neg(x, x);
  481.                             (* x := -x*)
  482.                         END ;
  483.                         S.class := 4; S.x := x
  484.                     END ;
  485.                     IF hex THEN S.class := 0 END
  486.                 ELSE (*decimal integer*)
  487.                     S.class := 3; k := 0;
  488.                     REPEAT k := k*10 + (ORD(d[j]) - 30H); INC(j) UNTIL j = i;
  489.                     IF neg THEN S.i := -k ELSE S.i := k END;
  490.                     IF hex THEN S.class := 0 ELSE S.class := 3 END
  491.                 END
  492.             ELSE S.class := 6;
  493.                 IF neg THEN S.c := "-" ELSE S.c := ch; Read(S, ch) END
  494.             END
  495.         END;
  496.         S.nextCh := ch
  497.     END Scan;
  498.     (** Writers **)
  499.     PROCEDURE OpenWriter* (VAR W: Writer);
  500.     BEGIN
  501.         NEW(W.buf);
  502.         OpenBuf(W.buf);
  503.         W.fnt := Fonts.Default; W.col := Display.white; W.voff := 0;
  504.         W.file := Files.New("");
  505.         Files.Set(W.rider, W.file, 0);
  506.     END OpenWriter;
  507.     PROCEDURE SetFont* (VAR W: Writer; fnt: Fonts.Font);
  508.     BEGIN W.fnt := fnt
  509.     END SetFont;
  510.     PROCEDURE SetColor* (VAR W: Writer; col: SHORTINT);
  511.     BEGIN W.col := col
  512.     END SetColor;
  513.     PROCEDURE SetOffset* (VAR W: Writer; voff: SHORTINT);
  514.     BEGIN W.voff := voff
  515.     END SetOffset;
  516.     PROCEDURE Write* (VAR W: Writer; ch: CHAR);
  517.         VAR u, un: Run; p: Piece;
  518.     BEGIN Files.Write(W.rider, ch); INC(W.buf.len); un := W.buf.head; u := un.prev;
  519.         IF (u IS Piece) & (u(Piece).file = W.file) & (u.fnt.name = W.fnt.name) & (u.col = W.col) & (u.voff = W.voff)
  520.         & ~u(Piece).ascii THEN (* << *)
  521.             INC(u.len)
  522.         ELSE NEW(p); u.next := p; p.prev := u; p.next := un; un.prev := p;
  523.             p.len := 1; p.fnt := W.fnt; p.col := W.col; p.voff := W.voff;
  524.             p.file := W.file; p.org := Files.Length(W.file) - 1; p.ascii := FALSE    (* << *)
  525.         END
  526.     END Write;
  527.     PROCEDURE WriteElem* (VAR W: Writer; e: Elem);
  528.         VAR u, un: Run;
  529.     BEGIN
  530.         IF e.base # NIL THEN HALT(99) END;
  531.         INC(W.buf.len); e.len := 1; e.fnt := W.fnt; e.col := W.col; e.voff := W.voff;
  532.         un := W.buf.head; u := un.prev; u.next := e; e.prev := u; e.next := un; un.prev := e
  533.     END WriteElem;
  534.     PROCEDURE WriteLn* (VAR W: Writer);
  535.     BEGIN Write(W, CR)
  536.     END WriteLn;
  537.     PROCEDURE WriteString* (VAR W: Writer; s: ARRAY OF CHAR);
  538.         VAR i: INTEGER;
  539.     BEGIN i := 0;
  540.         WHILE s[i] >= " " DO Write(W, s[i]); INC(i) END
  541.     END WriteString;
  542.     PROCEDURE WriteInt* (VAR W: Writer; x, n: LONGINT);
  543.         VAR i: INTEGER; x0: LONGINT;
  544.             a: ARRAY 11 OF CHAR;
  545.     BEGIN i := 0;
  546.         IF x < 0 THEN
  547.             IF x = MIN(LONGINT) THEN WriteString(W, " -2147483648"); RETURN
  548.             ELSE DEC(n); x0 := -x
  549.             END
  550.         ELSE x0 := x
  551.         END;
  552.         REPEAT
  553.             a[i] := CHR(x0 MOD 10 + 30H); x0 := x0 DIV 10; INC(i)
  554.         UNTIL x0 = 0;
  555.         WHILE n > i DO Write(W, " "); DEC(n) END;
  556.         IF x < 0 THEN Write(W, "-") END;
  557.         REPEAT DEC(i); Write(W, a[i]) UNTIL i = 0
  558.     END WriteInt;
  559.     PROCEDURE WriteHex* (VAR W: Writer; x: LONGINT);
  560.         VAR i: INTEGER; y: LONGINT;
  561.             a: ARRAY 10 OF CHAR;
  562.     BEGIN i := 0; Write(W, " ");
  563.         REPEAT y := x MOD 10H;
  564.             IF y < 10 THEN a[i] := CHR(y + 30H) ELSE a[i] := CHR(y + 37H) END;
  565.             x := x DIV 10H; INC(i)
  566.         UNTIL i = 8;
  567.         REPEAT DEC(i); Write(W, a[i]) UNTIL i = 0
  568.     END WriteHex;
  569.     PROCEDURE WriteReal* (VAR W: Writer; x: REAL; n: INTEGER);
  570.     VAR e: INTEGER; x0: REAL;
  571.             d: ARRAY maxD OF CHAR;
  572.             Dum: REAL;
  573.     BEGIN e := Reals.Expo(x);
  574.         IF e = 0 THEN
  575.             WriteString(W, "  0");
  576.             REPEAT Write(W, " "); DEC(n) UNTIL n <= 3
  577.         ELSIF e = 255 THEN
  578.             WriteString(W, " NaN");
  579.             WHILE n > 4 DO Write(W, " "); DEC(n) END
  580.         ELSE
  581.             IF n <= 9 THEN n := 3 ELSE DEC(n, 6) END;
  582.             REPEAT Write(W, " "); DEC(n) UNTIL n <= 8;
  583.             (*there are 2 < n <= 8 digits to be written*)
  584.             IF AmigaMath.Tst(x) < 0 THEN Write(W, "-"); AmigaMath.Neg(x, x) ELSE Write(W, " ") END;
  585.             (* IF x < 0.0 THEN Write(W, "-"); x := -x ELSE Write(W, " ") END; *)
  586.             e := (e - 127) * 77  DIV 256;
  587.             IF e >= 0 THEN
  588.                 Ten(e, Dum);
  589.                 AmigaMath.Div(x, Dum, x)
  590.             ELSE
  591.                 Ten(-e, Dum);
  592.                 AmigaMath.Mul(x, Dum, x)
  593.             END;
  594.             (* IF e >= 0 THEN x := x / Reals.Ten(e) ELSE x := Reals.Ten(-e) * x END; *)
  595.             IF AmigaMath.Cmp(x, 10.0) >= 0 THEN AmigaMath.Mul(x, 0.1, x); INC(e) END;
  596.             (* IF x >= 10.0 THEN x := 0.1*x; INC(e) END; *)
  597.             Ten(n-1, x0);
  598.             AmigaMath.Mul(x0, x, Dum);
  599.             AmigaMath.Add(Dum, 0.5, x);
  600.             (* x0 := Reals.Ten(n-1); x := x0*x + 0.5; *)
  601.             AmigaMath.Mul(10.0, x0, Dum);
  602.             IF AmigaMath.Cmp(x, Dum) >= 0 THEN AmigaMath.Mul(x, 0.1, x); INC(e) END;
  603.             (* IF x >= 10.0*x0 THEN x := x*0.1; INC(e) END; *)
  604.             Reals.Convert(x, n, d);
  605.             DEC(n); Write(W, d[n]); Write(W, ".");
  606.             REPEAT DEC(n); Write(W, d[n]) UNTIL n = 0;
  607.             Write(W, "E");
  608.             IF e < 0 THEN Write(W, "-"); e := -e ELSE Write(W, "+") END;
  609.             Write(W, CHR(e DIV 10 + 30H)); Write(W, CHR(e MOD 10 + 30H))
  610.         END
  611.     END WriteReal;
  612.     PROCEDURE WriteRealFix* (VAR W: Writer; x: REAL; n, k: INTEGER);
  613.         VAR e, i: INTEGER; sign: CHAR; x0: REAL;
  614.             d: ARRAY maxD OF CHAR;
  615.             Dum: REAL;
  616.         PROCEDURE seq(ch: CHAR; n: INTEGER);
  617.         BEGIN WHILE n > 0 DO Write(W, ch); DEC(n) END
  618.         END seq;
  619.         PROCEDURE dig(n: INTEGER);
  620.         BEGIN
  621.             WHILE n > 0 DO
  622.                 DEC(i); Write(W, d[i]); DEC(n)
  623.             END
  624.         END dig;
  625.     BEGIN e := Reals.Expo(x);
  626.         IF k < 0 THEN k := 0 END;
  627.         IF e = 0 THEN seq(" ", n-k-2); Write(W, "0"); seq(" ", k+1)
  628.         ELSIF e = 255 THEN WriteString(W, " NaN"); seq(" ", n-4)
  629.         ELSE e := (e - 127) * 77 DIV 256;
  630.             IF AmigaMath.Cmp(x,0.0) < 0 THEN sign := "-"; AmigaMath.Neg(x, x) ELSE sign := " " END;
  631.             (* IF x < 0 THEN sign := "-"; x := -x ELSE sign := " " END; *)
  632.             IF e >= 0 THEN  (*x >= 1.0,  77/256 = log 2*)
  633.                 Ten(e, Dum);
  634.                 AmigaMath.Div(x, Dum, x)
  635.             ELSE (*x < 1.0*)
  636.                 Ten(-e, Dum);
  637.                 AmigaMath.Mul(x, Dum, x)
  638.             END;
  639.             (* IF e >= 0 THEN  (*x >= 1.0,  77/256 = log 2*) x := x/Reals.Ten(e)
  640.                 ELSE (*x < 1.0*) x := Reals.Ten(-e) * x
  641.             END; *)
  642.             IF AmigaMath.Cmp(x, 10.0) >= 0 THEN AmigaMath.Mul(x, 0.1, x); INC(e) END;
  643.             (* IF x >= 10.0 THEN x := 0.1*x; INC(e) END; *)
  644.             (* 1 <= x < 10 *)
  645.             IF k+e >= maxD-1 THEN k := maxD-1-e
  646.                 ELSIF k+e < 0 THEN k := -e; x := 0.0
  647.             END;
  648.             Ten(k+e, x0); AmigaMath.Mul(x, x0, x); AmigaMath.Add(x, 0.5, x);
  649.             (* x0 := Reals.Ten(k+e); x := x0*x + 0.5; *)
  650.             AmigaMath.Mul(10.0, x0, Dum);
  651.             IF AmigaMath.Cmp(x, Dum) >= 0 THEN INC(e) END;
  652.             (* IF x >= 10.0*x0 THEN INC(e) END; *)
  653.             (*e = no. of digits before decimal point*)
  654.             INC(e); i := k+e; Reals.Convert(x, i, d);
  655.             IF e > 0 THEN
  656.                 seq(" ", n-e-k-2); Write(W, sign); dig(e);
  657.                 Write(W, "."); dig(k)
  658.             ELSE seq(" ", n-k-3);
  659.                 Write(W, sign); Write(W, "0"); Write(W, ".");
  660.                 seq("0", -e); dig(k+e)
  661.             END
  662.         END
  663.     END WriteRealFix;
  664.     PROCEDURE WriteRealHex* (VAR W: Writer; x: REAL);
  665.         VAR i: INTEGER;
  666.             d: ARRAY 8 OF CHAR;
  667.     BEGIN Reals.ConvertH(x, d); i := 0;
  668.         REPEAT Write(W, d[i]); INC(i) UNTIL i = 8
  669.     END WriteRealHex;
  670.     PROCEDURE WriteLongReal* (VAR W: Writer; x: LONGREAL; n: INTEGER);
  671.         CONST maxD = 16;
  672.         VAR e: INTEGER; x0: LONGREAL;
  673.             d: ARRAY maxD OF CHAR;
  674.             DumL: LONGREAL;
  675.     BEGIN e := Reals.ExpoL(x);
  676.         IF e = 0 THEN
  677.             WriteString(W, "  0");
  678.             REPEAT Write(W, " "); DEC(n) UNTIL n <= 3
  679.         ELSIF e = 2047 THEN
  680.             WriteString(W, " NaN");
  681.             WHILE n > 4 DO Write(W, " "); DEC(n) END
  682.         ELSE
  683.             IF n <= 10 THEN n := 3 ELSE DEC(n, 7) END;
  684.             REPEAT Write(W, " "); DEC(n) UNTIL n <= maxD;
  685.             (*there are 2 <= n <= maxD digits to be written*)
  686.             IF AmigaMathL.Tst(x) < 0 THEN Write(W, "-"); AmigaMathL.Neg(x, x) ELSE Write(W, " ") END;
  687.             (* IF x < 0 THEN Write(W, "-"); x := -x ELSE Write(W, " ") END; *)
  688.             e := SHORT(LONG(e - 1023) * 77 DIV 256);
  689.             IF e >= 0 THEN 
  690.                 TenL(e, DumL);
  691.                 AmigaMathL.Div(x, DumL, x);
  692.             ELSE
  693.                 TenL(-e, x);
  694.             END;
  695.             (* IF e >= 0 THEN x := x / Reals.TenL(e) ELSE x := Reals.TenL(-e) * x END ; *)
  696.             IF AmigaMathL.Cmp(x, 10.0D0) >= 0 THEN AmigaMathL.Mul(x,  0.1D0, x); INC(e) END ;
  697.             (* IF x >= 10.0D0 THEN x := 0.1D0 * x; INC(e) END ; *)
  698.             TenL(n-1, x0); AmigaMathL.Mul(x0, x, DumL); AmigaMathL.Add(DumL, 0.5D0, x);
  699.             (* x0 := Reals.TenL(n-1); x := x0*x + 0.5D0; *)
  700.             AmigaMathL.Mul(10.0D0, x0, DumL);
  701.             IF AmigaMathL.Cmp(x, DumL) >= 0 THEN AmigaMathL.Mul(x, 0.1D0, x); INC(e) END ;
  702.             (* IF x >= 10.0D0*x0 THEN x := 0.1D0 * x; INC(e) END ; *)
  703.             Reals.ConvertL(x, n, d);
  704.             DEC(n); Write(W, d[n]); Write(W, ".");
  705.             REPEAT DEC(n); Write(W, d[n]) UNTIL n = 0;
  706.             Write(W, "D");
  707.             IF e < 0 THEN Write(W, "-"); e := -e ELSE Write(W, "+") END;
  708.             Write(W, CHR(e DIV 100 + 30H)); e := e MOD 100;
  709.             Write(W, CHR(e DIV 10 + 30H));
  710.             Write(W, CHR(e MOD 10 + 30H))
  711.         END
  712.     END WriteLongReal;
  713.     PROCEDURE WriteLongRealHex* (VAR W: Writer; x: LONGREAL);
  714.         VAR i: INTEGER;
  715.             d: ARRAY 16 OF CHAR;
  716.     BEGIN Reals.ConvertHL(x, d); i := 0;
  717.         REPEAT Write(W, d[i]); INC(i) UNTIL i = 16
  718.     END WriteLongRealHex;
  719.     PROCEDURE WriteDate* (VAR W: Writer; t, d: LONGINT);
  720.         PROCEDURE WritePair(ch: CHAR; x: LONGINT);
  721.         BEGIN Write(W, ch);
  722.           Write(W, CHR(x DIV 10 + 30H)); Write(W, CHR(x MOD 10 + 30H))
  723.         END WritePair;
  724.     BEGIN
  725.         WritePair(" ", d MOD 32); WritePair(".", d DIV 32 MOD 16); WritePair(".", d DIV 512 MOD 128);
  726.         WritePair(" ", t DIV 4096 MOD 32); WritePair(":", t DIV 64 MOD 64); WritePair(":", t MOD 64)
  727.     END WriteDate;
  728.     (** Text Filing **)
  729.     PROCEDURE Load0 (VAR r: Files.Rider; T: Text);
  730.         VAR u, un: Run; p: Piece; e: Elem;
  731.             org, pos, hlen, plen: LONGINT; ecnt, fno, fcnt, col, voff: SHORTINT;
  732.             f: Files.File;
  733.             msg: FileMsg;
  734.             mods, procs: ARRAY 64, 32 OF CHAR;
  735.             name: ARRAY 32 OF CHAR;
  736.             fnts: ARRAY 32 OF Fonts.Font;
  737.         PROCEDURE LoadElem (VAR r: Files.Rider; pos, span: LONGINT; VAR e: Elem);
  738.             VAR M: Modules.Module; Cmd: Modules.Command; a: Alien;
  739.                 org, ew, eh: LONGINT; eno: SHORTINT;
  740.         BEGIN new := NIL;
  741.             Files.ReadLInt(r, ew); Files.ReadLInt(r, eh); Files.Read(r, eno);
  742.             IF eno > ecnt THEN ecnt := eno; Files.ReadString(r, mods[eno]); Files.ReadString(r, procs[eno]) END;
  743.             org := Files.Pos(r); M := Modules.ThisMod(mods[eno]);
  744.             IF M # NIL THEN Cmd := Modules.ThisCommand(M, procs[eno]);
  745.                 IF Cmd # NIL THEN Cmd END
  746.             END;
  747.             e := new;
  748.             IF e # NIL THEN e.W := ew; e.H := eh; e.base := T;
  749.                 msg.pos := pos; e.handle(e, msg);
  750.                 IF Files.Pos(r) # org + span THEN e := NIL END
  751.             END;
  752.             IF e = NIL THEN Files.Set(r, f, org + span);
  753.                 NEW(a); a.W := ew; a.H := eh; a.handle := HandleAlien; a.base := T;
  754.                 a.file := f; a.org := org; a.span := span;
  755.                 COPY(mods[eno], a.mod); COPY(procs[eno], a.proc);
  756.                 e := a
  757.             END
  758.         END LoadElem;
  759.     BEGIN pos := Files.Pos(r); f := Files.Base(r);
  760.         NEW(u); u.len := MAX(LONGINT); (*u.fnt := Fonts.Default;*)u.fnt := NIL; u.col := Display.white;
  761.         T.head := u; ecnt := 0; fcnt := 0;
  762.         msg.id := load; msg.r := r;
  763.         Files.ReadLInt(msg.r, hlen); (*!!!org := pos + hlen;*) org := pos -2 + hlen; pos := org; Files.Read(msg.r, fno);
  764.         WHILE fno # 0 DO
  765.             IF fno > fcnt THEN fcnt := fno; Files.ReadString(msg.r, name); fnts[fno] := Fonts.This(name) END;
  766.             Files.Read(msg.r, col); Files.Read(msg.r, voff); Files.ReadLInt(msg.r, plen);
  767.             IF plen > 0 THEN NEW(p); p.file := f; p.org := pos; p.ascii := FALSE; un := p; un.len := plen    (* << *)
  768.             ELSE LoadElem(msg.r, pos - org, -plen, e); un := e; un.len := 1
  769.             END;
  770.             un.fnt := fnts[fno]; un.col := col; un.voff := voff;
  771.             INC(pos, un.len); u.next := un; un.prev := u; u := un; Files.Read(msg.r, fno)
  772.         END;
  773.         u.next := T.head; T.head.prev := u; T.cache := T.head; T.corg := 0;
  774.         Files.ReadLInt(msg.r, T.len); Files.Set(r, f, Files.Pos(msg.r) + T.len)
  775.     END Load0;
  776.     PROCEDURE Load* (VAR r: Files.Rider; T: Text);
  777.         CONST oldTag = -4095;
  778.         VAR tag: INTEGER;
  779.     BEGIN
  780.         (* for compatibility inner text tags are checked and skipped; remove this in a later version *)
  781.         Files.ReadInt(r, tag); IF tag # oldTag THEN Files.Set(r, Files.Base(r), Files.Pos(r)-2) END;
  782.         Load0(r, T)
  783.     END Load;
  784.     PROCEDURE Open* (T: Text; name: ARRAY OF CHAR);
  785.         VAR f: Files.File; r: Files.Rider; u: Run; p: Piece; tag, version: CHAR;
  786.     BEGIN f := Files.Old(name);
  787.         IF f = NIL THEN f := Files.New("") END;
  788.         Files.Set(r, f, 0); Files.Read(r, tag); Files.Read(r, version);
  789.         IF (tag = textTag) OR (tag = 01X) & (version = textTag) THEN Load0(r, T)
  790.         ELSE NEW(u); u.len := MAX(LONGINT); u.fnt := NIL; u.col := Display.white;
  791.             T.len := Files.Length(f);
  792.             IF T.len > 0 THEN NEW(p); p.len := T.len; p.fnt := Fonts.Default;
  793.                 p.col := Display.white; p.voff := 0; p.file := f; p.org := 0; p.ascii := TRUE;    (* << *)
  794.                 u.next := p; u.prev := p; p.next := u; p.prev := u
  795.             ELSE u.next := u; u.prev := u
  796.             END;
  797.             T.head := u; T.cache := T.head; T.corg := 0
  798.         END
  799.     END Open;
  800.     PROCEDURE Store* (VAR r: Files.Rider; T: Text);
  801.         VAR r1: Files.Rider; u, un: Run; e: Elem; org, pos, delta, hlen, rlen: LONGINT; ecnt, fno, fcnt: SHORTINT;
  802.             ch: CHAR;    (* << *)
  803.             msg: FileMsg; iden: IdentifyMsg;
  804.             mods, procs: ARRAY 64, 32 OF CHAR;
  805.             fnts: ARRAY 32 OF Fonts.Font;
  806.             block: ARRAY 1024 OF CHAR;
  807.         PROCEDURE StoreElem (VAR r: Files.Rider; pos: LONGINT; e: Elem);
  808.             VAR r1: Files.Rider; org, span: LONGINT; eno: SHORTINT;
  809.         BEGIN COPY(iden.mod, mods[ecnt]); COPY(iden.proc, procs[ecnt]); eno := 1;
  810.             WHILE (mods[eno] # iden.mod) OR (procs[eno] # iden.proc) DO INC(eno) END;
  811.             Files.Set(r1, Files.Base(r), Files.Pos(r));
  812.             Files.WriteLInt(r, 0); Files.WriteLInt(r, 0); Files.WriteLInt(r, 0); (*fixup slot*)
  813.             Files.Write(r, eno);
  814.             IF eno = ecnt THEN INC(ecnt); Files.WriteString(r, iden.mod); Files.WriteString(r, iden.proc) END;
  815.             msg.pos := pos; org := Files.Pos(r); e.handle(e, msg); span := Files.Pos(r) - org;
  816.             Files.WriteLInt(r1, -span); Files.WriteLInt(r1, e.W); Files.WriteLInt(r1, e.H) (*fixup*)
  817.         END StoreElem;
  818.     BEGIN
  819.         org := Files.Pos(r); msg.id := store; msg.r := r; Files.WriteLInt(msg.r, 0); (*fixup slot*)
  820.         u := T.head.next; pos := 0; delta := 0; fcnt := 1; ecnt := 1;
  821.         WHILE u # T.head DO
  822.             IF u IS Elem THEN iden.mod[0] := 0X; u(Elem).handle(u(Elem), iden) ELSE iden.mod[0] := 1X END;
  823.             IF iden.mod[0] # 0X THEN
  824.                 fnts[fcnt] := u.fnt; fno := 1;
  825.                 WHILE fnts[fno].name # u.fnt.name DO INC(fno) END;
  826.                 Files.Write(msg.r, fno);
  827.                 IF fno = fcnt THEN INC(fcnt); Files.WriteString(msg.r, u.fnt.name) END;
  828.                 Files.Write(msg.r, u.col); Files.Write(msg.r, u.voff)
  829.             END;
  830.             IF u IS Piece THEN rlen := u.len; un := u.next;
  831.                 WHILE (un IS Piece) & (un.fnt = u.fnt) & (un.col = u.col) & (un.voff = u.voff) DO
  832.                     INC(rlen, un.len); un := un.next
  833.                 END;
  834.                 Files.WriteLInt(msg.r, rlen); INC(pos, rlen); u := un
  835.             ELSIF iden.mod[0] # 0X THEN StoreElem(msg.r, pos, u(Elem)); INC(pos); u := u.next
  836.             ELSE INC(delta); u := u.next
  837.             END
  838.         END;
  839.         Files.Write(msg.r, 0); Files.WriteLInt(msg.r, T.len - delta);
  840.         (*!!!hlen := Files.Pos(msg.r) - org;*) hlen := Files.Pos(msg.r) - org + 2;
  841.         Files.Set(r1, Files.Base(msg.r), org); Files.WriteLInt(r1, hlen); (*fixup*)
  842.         u := T.head.next;
  843.         WHILE u # T.head DO
  844.             IF u IS Piece THEN
  845.                 WITH u: Piece DO
  846.                     IF u.ascii THEN Files.Set(r1, u.file, u.org); delta := u.len;    (* << LF to CR *)
  847.                         WHILE delta > 0 DO Files.Read(r1, ch); DEC(delta);
  848.                             IF ch = 0AX THEN Files.Write(msg.r, CR) ELSE Files.Write(msg.r, ch) END
  849.                         END
  850.                     ELSE Files.Set(r1, u.file, u.org); delta := u.len;
  851.                         WHILE delta > LEN(block) DO Files.ReadBytes(r1, block, LEN(block));
  852.                             Files.WriteBytes(msg.r, block, LEN(block)); DEC(delta, LEN(block))
  853.                         END;
  854.                         Files.ReadBytes(r1, block, delta); Files.WriteBytes(msg.r, block, delta)
  855.                     END
  856.                 END
  857.             ELSE iden.mod[0] := 0X; u(Elem).handle(u(Elem), iden);
  858.                 IF iden.mod[0] # 0X THEN Files.Write(msg.r, ElemChar) END
  859.             END;
  860.             u := u.next
  861.         END;
  862.         r := msg.r;
  863.     END Store;
  864.     PROCEDURE Close* (T: Text; name: ARRAY OF CHAR);
  865.         VAR f: Files.File; r: Files.Rider; i, res: INTEGER; bak: ARRAY 64 OF CHAR;
  866.     BEGIN
  867.         f := Files.New(name); Files.Set(r, f, 0); Files.Write(r, textTag); Files.Write(r, version); Store(r, T);
  868.         i := 0; WHILE name[i] # 0X DO INC(i) END;
  869.         COPY(name, bak); bak[i] := "."; bak[i+1] := "B"; bak[i+2] := "a"; bak[i+3] := "k"; bak[i+4] := 0X;
  870.         Files.Rename(name, bak, res); Files.Register(f)
  871.     END Close;
  872. BEGIN del := NIL
  873. END Texts.
  874.